
 <!DOCTYPE HTML>
<html >
<head>
  <meta charset="UTF-8">
  
    <title>聊一聊幂等 | Typhoon&#39;s code space</title>
    <meta name="viewport" content="width=device-width, initial-scale=1,user-scalable=no">
    
    <meta name="author" content="typhoon">
    

    
    <meta name="description" content="幂等是什么？百度上给出的解释如下： 幂等（idempotent、idempotence）是一个数学与计算机学概念，常见于抽象代数中。 在编程中一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同。">
<meta name="keywords" content="幂等">
<meta property="og:type" content="article">
<meta property="og:title" content="聊一聊幂等">
<meta property="og:url" content="http://scorpioaeolus.oschina.io/2018/11/30/聊一聊幂等/index.html">
<meta property="og:site_name" content="Typhoon&#39;s code space">
<meta property="og:description" content="幂等是什么？百度上给出的解释如下： 幂等（idempotent、idempotence）是一个数学与计算机学概念，常见于抽象代数中。 在编程中一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同。">
<meta property="og:image" content="http://scorpioaeolus.oschina.io/typhoon/2018/11/30/聊一聊幂等/id1.jpg">
<meta property="og:image" content="http://scorpioaeolus.oschina.io/typhoon/2018/11/30/聊一聊幂等/id2.jpg">
<meta property="og:updated_time" content="2018-12-02T07:07:25.054Z">
<meta name="twitter:card" content="summary">
<meta name="twitter:title" content="聊一聊幂等">
<meta name="twitter:description" content="幂等是什么？百度上给出的解释如下： 幂等（idempotent、idempotence）是一个数学与计算机学概念，常见于抽象代数中。 在编程中一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同。">
<meta name="twitter:image" content="http://scorpioaeolus.oschina.io/typhoon/2018/11/30/聊一聊幂等/id1.jpg">

    
    <link rel="alternative" href="/atom.xml" title="Typhoon&#39;s code space" type="application/atom+xml">
    
    
    <link rel="icon" href="/typhoon/img/java.png">
    
    
    <link rel="apple-touch-icon" href="/typhoon/img/jacman.jpg">
    <link rel="apple-touch-icon-precomposed" href="/typhoon/img/jacman.jpg">
    
    <link rel="stylesheet" href="/typhoon/css/style.css">
</head>

  <body>
    <header>
      
<div>
		
			<div id="imglogo">
				<a href="/typhoon/"><img src="/typhoon/img/java.png" alt="Typhoon&#39;s code space" title="Typhoon&#39;s code space"/></a>
			</div>
			
			<div id="textlogo">
				<h1 class="site-name"><a href="/typhoon/" title="Typhoon&#39;s code space">Typhoon&#39;s code space</a></h1>
				<h2 class="blog-motto"></h2>
			</div>
			<div class="navbar"><a class="navbutton navmobile" href="#" title="Menu">
			</a></div>
			<nav class="animated">
				<ul>
					<ul>
					 
						<li><a href="/typhoon/">主页</a></li>
					
						<li><a href="/typhoon/archives">归档</a></li>
					
						<li><a href="/typhoon/about">关于</a></li>
					
					<li>
 					
					<form class="search" action="//google.com/search" method="get" accept-charset="utf-8">
						<label>Search</label>
						<input type="search" id="search" name="q" autocomplete="off" maxlength="20" placeholder="Search" />
						<input type="hidden" name="q" value="site:scorpioaeolus.oschina.io">
					</form>
					
					</li>
				</ul>
			</nav>			
</div>
    </header>
    <div id="container">
      <div id="main" class="post" itemscope itemprop="blogPost">
  
	<article itemprop="articleBody"> 
		<header class="article-info clearfix">
  <h1 itemprop="name">
    
      <a href="/typhoon/2018/11/30/聊一聊幂等/" title="聊一聊幂等" itemprop="url">聊一聊幂等</a>
  </h1>
  <p class="article-author">By
       
		<a href="/typhoon/about" title="typhoon" target="_blank" itemprop="author">typhoon</a>
		
  <p class="article-time">
    <time datetime="2018-11-30T13:02:39.000Z" itemprop="datePublished"> Published 2018-11-30</time>
    
  </p>
</header>
	<div class="article-content">
		
		<div id="toc" class="toc-article">
			<strong class="toc-title">Contents</strong>
		
			<ol class="toc"><li class="toc-item toc-level-5"><a class="toc-link" href="#一、编程中的幂等"><span class="toc-number">1.</span> <span class="toc-text">一、编程中的幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#二、接口幂等"><span class="toc-number">2.</span> <span class="toc-text">二、接口幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#三、消息幂等"><span class="toc-number">3.</span> <span class="toc-text">三、消息幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#四、数据库幂等"><span class="toc-number">4.</span> <span class="toc-text">四、数据库幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#总结"><span class="toc-number">5.</span> <span class="toc-text">总结</span></a></li></ol>
		
		</div>
		
		<pre><code>    幂等是什么？百度上给出的解释如下：
幂等（idempotent、idempotence）是一个数学与计算机学概念，常见于抽象代数中。
在编程中一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同。
</code></pre><a id="more"></a>
<pre><code>幂等函数，或幂等方法，是指可以使用相同参数重复执行，并能获得相同结果的函数。
这些函数不会影响系统状态，也不用担心重复执行会对系统造成改变。例如，“setTrue()”
函数就是一个幂等函数,无论多次执行，其结果都是一样的.更复杂的操作幂等保证是利用
唯一交易号(流水号)实现。
</code></pre><h5 id="一、编程中的幂等"><a href="#一、编程中的幂等" class="headerlink" title="一、编程中的幂等"></a>一、编程中的幂等</h5><p><strong>概念</strong></p>
<pre><code>    在我们日常开发和业务实现中，对于相同的参数输入，多次调用相同的功能，对资源的影响是一样的，也就是
一次和多次请求某一个资源应该具有同样的副作用。
</code></pre><p><strong>幂等解决的问题</strong></p>
<ul>
<li>表单重复提交；重复推送数据导致多次更新后端资源导致数据不一致问题</li>
<li>RPC超时重试；服务被多次调用导致数据不一致问题</li>
<li>SQL多次执行；程序问题导致sql多次调用带来数据不一致问题</li>
</ul>
<p><strong>常见的幂等场景</strong></p>
<pre><code>我们在研发工作中，需要考虑幂等的场景也比较多，常见的就是接口幂等、消息幂等和数据库幂等等。
</code></pre><h5 id="二、接口幂等"><a href="#二、接口幂等" class="headerlink" title="二、接口幂等"></a>二、接口幂等</h5><pre><code>接口幂等又常分为http接口幂等和RPC接口幂等。
</code></pre><p><strong>HTTP接口幂等</strong></p>
<pre><code>我们常用的http接口对应的请求方式中：
</code></pre><ul>
<li>GET是幂等的，get方式是从服务器端获取资源，是单纯的查询操作，对服务端资源没有更新，所以是幂等的。</li>
<li>HEAD是幂等的，似于get请求，只不过返回的响应中没有具体的内容，用于获取报头。</li>
<li>POST一般是非幂等的，用于表单提交向服务端新增数据。</li>
<li>PUT 一般也是幂等的，用于更新服务端资源。</li>
<li><p>DELETE 一般也是幂等的，用于删除服务端资源。</p>
<pre><code>其他请求方式我们平时基本很少用到，这里不再一一列举。
</code></pre><p><strong>RPC接口幂等</strong></p>
<pre><code>RPC接口用于领域设计后的功能拆分，调用是跨进程的，对于RPC接口中的幂等，其实是对于外部调用超时重试，或者同样参数多次调用同一个接口，
</code></pre><p>  要保证对服务端资源的影响和一次调用是一样的，我们经常遇到的情况是状态机的变更，比如客户端调用RPC服务完结退款的状态，那么多次调用要保证<br>  和一次调用一样，退款状态只能被修改一次，并且最终的状态是完结。</p>
</li>
</ul>
<h5 id="三、消息幂等"><a href="#三、消息幂等" class="headerlink" title="三、消息幂等"></a>三、消息幂等</h5><pre><code>    对于消息幂等，可能大家对其了解不是很多，我们应用中基本都会使用消息队列，那么肯定会涉及到消息投递和消息接收，消息幂等也要分两个维度来
分析：
</code></pre><p><strong>消息投递幂等</strong></p>
<pre><code>所谓消息投递幂等，就是同一条消息只能被投递一次，对于消息broker来说，就算同一条消息投递多次，我也只存储一条。我们举个例子来说明一下：
</code></pre><img src="/typhoon/2018/11/30/聊一聊幂等/id1.jpg" alt="idempotent" title="idempotent">
<ul>
<li>①消息发送者，尝试发送一条消息到消息broker</li>
<li>②消息broker收到消息后理论上要给一个响应结果给发送者，但是这个响应可能丢失了</li>
<li><p>③对于②中的响应丢失或者没有响应，消息发送者会认为没有发送成功，重复投递消息</p>
<pre><code>问题就在于响应丢失重复投递，有可能消息broker已经成功接收消息并且存储了，重复投递的消息有可能被消息broker接收并存储，导致broker接收了
</code></pre><p>  两条或者两条以上的相同消息，也就会导致消息接收方接收到多条相同的消息，在业务场景中可能造成业务异常。</p>
<pre><code>对于这种重复投递的消息，消息broker层可以对每一条消息生成一个唯一的code，有重复消息过来的时候，生成的code也会相同，如果发现相同就丢弃。
</code></pre><p><strong>消息接收幂等</strong></p>
<pre><code>消息接收幂等，是消息broker中的同一条消息只能被consumer接收处理一次，就算broker推送多次，也只能消费一条。同样举个例子来说明一下：
</code></pre><img src="/typhoon/2018/11/30/聊一聊幂等/id2.jpg" alt="idempotent" title="idempotent">
</li>
<li><p>①消息broker尝试向consumer推送一条消息</p>
</li>
<li>②消息consumer接收到消息后，向broker发送响应结果，但是丢失</li>
<li>③消息broker由于没有接收到②中的响应结果，认为consumer没有收到消息，会重复推送<pre><code>消息broker重复推送相同消息的时候，有可能consumer已经接受成功并处理了业务，再次收到消息会导致consumer重复处理逻辑，假如在资金相关的场景
</code></pre>  出现这种情况，会导致重复出账造成资损。<pre><code>对于消息broker重复推送的消息，consumer要对每一条消息生成唯一id或者code，如果发现重复消息，直接丢弃。
</code></pre></li>
</ul>
<h5 id="四、数据库幂等"><a href="#四、数据库幂等" class="headerlink" title="四、数据库幂等"></a>四、数据库幂等</h5><pre><code>    所谓数据库幂等，也就是我们对DB层的操作幂等，说人话就是保证我们业务操作sql是幂等的，其实就是我们同一条sql执行多次和执行一次的效果是一样的，
就拿CRUD来说，有些是幂等的，有些不是幂等但是可以通过调整转换成幂等的：
</code></pre><p><strong>查询(Retrieve)</strong></p>
<pre><code>对于数据库查询，只是单穿的从数据库获取资源，不会更新数据，所以是幂等的。
</code></pre><p><strong>创建(Create)</strong></p>
<pre><code>对于新增数据操作，很多时候是非幂等的，执行一次insert就会新增一条数据(id是自增主键)：
</code></pre><figure class="highlight java"><table><tr><td class="gutter"><pre><div class="line">1</div></pre></td><td class="code"><pre><div class="line"><span class="function">insert into <span class="title">User</span><span class="params">(id,name)</span> <span class="title">values</span><span class="params">(<span class="keyword">null</span>,<span class="string">'typhoon'</span>)</span></span>;</div></pre></td></tr></table></figure>
<pre><code>这条sql是非幂等的，每执行一次都会新增一条记录，但是通过改造我们可以将其变成幂等的：
</code></pre><figure class="highlight java"><table><tr><td class="gutter"><pre><div class="line">1</div></pre></td><td class="code"><pre><div class="line"><span class="function">insert into <span class="title">User</span><span class="params">(id,name)</span> <span class="title">values</span><span class="params">(<span class="number">100</span>,<span class="string">'typhoon'</span>)</span></span>;</div></pre></td></tr></table></figure>
<pre><code>这样就变成幂等的了，不使用自增序列，通过程序生成主键，这样重复执行，数据库也只会新增一条数据。
</code></pre><p><strong>更新(Update)</strong></p>
<pre><code>数据库更新操作，有些是幂等的，有些是非幂等的，典型的场景就是值递增，比如把小明的账户新增10000块：
</code></pre><figure class="highlight java"><table><tr><td class="gutter"><pre><div class="line">1</div></pre></td><td class="code"><pre><div class="line">update Account set money = money + <span class="number">10000</span> where name = <span class="string">'小明'</span>;</div></pre></td></tr></table></figure>
<pre><code>这条sql是非幂等的，如果由于上层程序有bug，导致该sql重复被调用，那么就赶紧找地方哭去吧。我们通过简单的调整就可以把上述sql给造成幂等的：
</code></pre><figure class="highlight java"><table><tr><td class="gutter"><pre><div class="line">1</div><div class="line">2</div><div class="line">3</div></pre></td><td class="code"><pre><div class="line">var currentMoney = select money from Account where name = <span class="string">'小明'</span>;</div><div class="line">var targetMoney = currentMoney + <span class="number">10000</span>;</div><div class="line">update Account set money = targetMoney where name = <span class="string">'小明'</span> and money = currentMoney;</div></pre></td></tr></table></figure>
<pre><code>这样上述的非幂等sql就改造成了幂等了，有两个关键的点需要注意：
</code></pre><ul>
<li>使用”=”代替”+”，避免多次执行导致多次增加。</li>
<li>增加条件money = currentMoney，避免并发的其他程序把金额扣了，导致这条sql把扣掉的钱又加上了。</li>
</ul>
<p><strong>删除(Delete)</strong></p>
<pre><code>带精确匹配的删除sql是幂等的，其他的基本是非幂等的，典型的就是范围删除：
</code></pre><figure class="highlight java"><table><tr><td class="gutter"><pre><div class="line">1</div></pre></td><td class="code"><pre><div class="line">delete from Table where id &gt; <span class="number">1000</span>;</div></pre></td></tr></table></figure>
<pre><code>    这个sql是非幂等的，在删除数据的同时，有并发程序在新增数据，那么就导致每一次执行delete都删除了数据，也就违反了执行多次和一次对资源的
副作用一样。经过改造把上述sql改成幂等的：
</code></pre><figure class="highlight java"><table><tr><td class="gutter"><pre><div class="line">1</div></pre></td><td class="code"><pre><div class="line"><span class="function">delete from Table where id <span class="title">in</span> <span class="params">(?,?,?)</span></span>;</div></pre></td></tr></table></figure>
<pre><code>这样把范围删除变成了精确匹配删除，也就变成了幂等sql。
</code></pre><h5 id="总结"><a href="#总结" class="headerlink" title="总结"></a>总结</h5><pre><code>    幂等对我们应用架构非常重要，大公司把幂等当做一个应用编码规范，可见其重要程度，不管是网络超时重试、
还是程序有bug导致的多次调用，幂等在很大程度上保护了我们的系统资源。
</code></pre>  
	</div>
		<footer class="article-footer clearfix">
<div class="article-catetags">

<div class="article-categories">
  <span></span>
  <a class="article-category-link" href="/typhoon/categories/架构/">架构</a>
</div>


  <div class="article-tags">
  
  <span></span> <a href="/typhoon/tags/幂等/">幂等</a>
  </div>

</div>



	<div class="article-share" id="share">
	
	  <div data-url="http://scorpioaeolus.oschina.io/2018/11/30/聊一聊幂等/" data-title="聊一聊幂等 | Typhoon&#39;s code space" data-tsina="null" class="share clearfix">
	  </div>
	
	</div>


</footer>

   	       
	</article>
	
<nav class="article-nav clearfix">
 
 <div class="prev" >
 <a href="/typhoon/2018/12/02/springboot自动配置/" title="springboot自动配置">
  <strong>上一篇：</strong><br/>
  <span>
  springboot自动配置</span>
</a>
</div>


<div class="next">
<a href="/typhoon/2018/11/25/springboot-统一异常拦截/"  title="springboot-统一异常拦截">
 <strong>下一篇：</strong><br/> 
 <span>springboot-统一异常拦截
</span>
</a>
</div>

</nav>

	


</div>  
      <div class="openaside"><a class="navbutton" href="#" title="Show Sidebar"></a></div>

  <div id="toc" class="toc-aside">
  <strong class="toc-title">Contents</strong>
 
 <ol class="toc"><li class="toc-item toc-level-5"><a class="toc-link" href="#一、编程中的幂等"><span class="toc-number">1.</span> <span class="toc-text">一、编程中的幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#二、接口幂等"><span class="toc-number">2.</span> <span class="toc-text">二、接口幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#三、消息幂等"><span class="toc-number">3.</span> <span class="toc-text">三、消息幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#四、数据库幂等"><span class="toc-number">4.</span> <span class="toc-text">四、数据库幂等</span></a></li><li class="toc-item toc-level-5"><a class="toc-link" href="#总结"><span class="toc-number">5.</span> <span class="toc-text">总结</span></a></li></ol>
 
  </div>

<div id="asidepart">
<div class="closeaside"><a class="closebutton" href="#" title="Hide Sidebar"></a></div>
<aside class="clearfix">

  


  
<div class="categorieslist">
	<p class="asidetitle">Categories</p>
		<ul>
		
		  
			<li><a href="/typhoon/categories/dubbo/" title="dubbo">dubbo<sup>8</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/java性能优化/" title="java性能优化">java性能优化<sup>17</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/java源码分析/" title="java源码分析">java源码分析<sup>1</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/motan/" title="motan">motan<sup>2</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/spring/" title="spring">spring<sup>11</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/springboot/" title="springboot">springboot<sup>28</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/springmvc/" title="springmvc">springmvc<sup>1</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/容器/" title="容器">容器<sup>1</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/工具/" title="工具">工具<sup>7</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/性能优化/" title="性能优化">性能优化<sup>5</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/架构/" title="架构">架构<sup>8</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/案例分析/" title="案例分析">案例分析<sup>13</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/缓存/" title="缓存">缓存<sup>4</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/设计模式/" title="设计模式">设计模式<sup>3</sup></a></li>
		  
		
		  
			<li><a href="/typhoon/categories/面试题/" title="面试题">面试题<sup>1</sup></a></li>
		  
		
		</ul>
</div>


  
<div class="tagslist">
	<p class="asidetitle">Tags</p>
		<ul class="clearfix">
		
			
				<li><a href="/typhoon/tags/springboot/" title="springboot">springboot<sup>28</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/java性能优化/" title="java性能优化">java性能优化<sup>9</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/限流/" title="限流">限流<sup>3</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/Spring5/" title="Spring5">Spring5<sup>3</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/设计模式/" title="设计模式">设计模式<sup>3</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/案例分析/" title="案例分析">案例分析<sup>2</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/性能优化/" title="性能优化">性能优化<sup>2</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/Spring/" title="Spring">Spring<sup>2</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/dubbo/" title="dubbo">dubbo<sup>2</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/分页/" title="分页">分页<sup>2</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/并发/" title="并发">并发<sup>2</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/异常拦截/" title="异常拦截">异常拦截<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/dubbo分布式事务/" title="dubbo分布式事务">dubbo分布式事务<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/自带图片/" title="自带图片">自带图片<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/svn导出mvn项目/" title="svn导出mvn项目">svn导出mvn项目<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/dubbo过滤器/" title="dubbo过滤器">dubbo过滤器<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/排序性能对比/" title="排序性能对比">排序性能对比<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/ListenableFuture/" title="ListenableFuture">ListenableFuture<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/PropertyChangeListener/" title="PropertyChangeListener">PropertyChangeListener<sup>1</sup></a></li>
			
		
			
				<li><a href="/typhoon/tags/jdbcTemplate高可用/" title="jdbcTemplate高可用">jdbcTemplate高可用<sup>1</sup></a></li>
			
		
		</ul>
</div>


  <div class="linkslist">
  <p class="asidetitle">Links</p>
    <ul>
        
          <li>
            
            	<a href="https://coderq.com" target="_blank" title="一个面向程序员交流分享的新一代社区">码农圈</a>
            
          </li>
        
          <li>
            
            	<a href="http://wuchong.me" target="_blank" title="Jark&#39;s Blog">Jark&#39;s Blog</a>
            
          </li>
        
    </ul>
</div>

  


  <div class="rsspart">
	<a href="/atom.xml" target="_blank" title="rss">RSS</a>
</div>

  <div class="weiboshow">
  <p class="asidetitle">Weibo</p>
    <iframe width="100%" height="119" class="share_self"  frameborder="0" scrolling="no" src="http://widget.weibo.com/weiboshow/index.php?language=&width=0&height=119&fansRow=2&ptype=1&speed=0&skin=9&isTitle=1&noborder=1&isWeibo=0&isFans=0&uid=null&verifier=b3593ceb&dpc=1"></iframe>
</div>


</aside>
</div>
    </div>
    <footer><div id="footer" >
	
	<div class="line">
		<span></span>
		<div class="author"></div>
	</div>
	
	
	<section class="info">
		<p> Hello ,I&#39;m a java coder. <br/>
			This is my blog,believe it or not.</p>
	</section>
	 
	<div class="social-font" class="clearfix">
		
		
		
		
		
		
		
		
		
		
		<a href="mailto:ScorpioAeolus@163.com" target="_blank" class="icon-email" title="Email Me"></a>
		
	</div>
			
		

		<p class="copyright">
		Powered by <a href="http://hexo.io" target="_blank" title="hexo">hexo</a> and Theme by <a href="https://github.com/wuchong/jacman" target="_blank" title="Jacman">Jacman</a> © 2019 
		
		<a href="/typhoon/about" target="_blank" title="typhoon">typhoon</a>
		
		
		</p>
</div>
</footer>
    <script src="/typhoon/js/jquery-2.0.3.min.js"></script>
<script src="/typhoon/js/jquery.imagesloaded.min.js"></script>
<script src="/typhoon/js/gallery.js"></script>
<script src="/typhoon/js/jquery.qrcode-0.12.0.min.js"></script>

<script type="text/javascript">
$(document).ready(function(){ 
  $('.navbar').click(function(){
    $('header nav').toggleClass('shownav');
  });
  var myWidth = 0;
  function getSize(){
    if( typeof( window.innerWidth ) == 'number' ) {
      myWidth = window.innerWidth;
    } else if( document.documentElement && document.documentElement.clientWidth) {
      myWidth = document.documentElement.clientWidth;
    };
  };
  var m = $('#main'),
      a = $('#asidepart'),
      c = $('.closeaside'),
      o = $('.openaside');
  c.click(function(){
    a.addClass('fadeOut').css('display', 'none');
    o.css('display', 'block').addClass('fadeIn');
    m.addClass('moveMain');
  });
  o.click(function(){
    o.css('display', 'none').removeClass('beforeFadeIn');
    a.css('display', 'block').removeClass('fadeOut').addClass('fadeIn');      
    m.removeClass('moveMain');
  });
  $(window).scroll(function(){
    o.css("top",Math.max(80,260-$(this).scrollTop()));
  });
  
  $(window).resize(function(){
    getSize(); 
    if (myWidth >= 1024) {
      $('header nav').removeClass('shownav');
    }else{
      m.removeClass('moveMain');
      a.css('display', 'block').removeClass('fadeOut');
      o.css('display', 'none');
      
      $('#toc.toc-aside').css('display', 'none');
        
    }
  });
});
</script>

<script type="text/javascript">
$(document).ready(function(){ 
  var ai = $('.article-content>iframe'),
      ae = $('.article-content>embed'),
      t  = $('#toc'),
      ta = $('#toc.toc-aside'),
      o  = $('.openaside'),
      c  = $('.closeaside');
  if(ai.length>0){
    ai.wrap('<div class="video-container" />');
  };
  if(ae.length>0){
   ae.wrap('<div class="video-container" />');
  };
  c.click(function(){
    ta.css('display', 'block').addClass('fadeIn');
  });
  o.click(function(){
    ta.css('display', 'none');
  });
  $(window).scroll(function(){
    ta.css("top",Math.max(140,320-$(this).scrollTop()));
  });
});
</script>


<script type="text/javascript">
$(document).ready(function(){ 
  var $this = $('.share'),
      url = $this.attr('data-url'),
      encodedUrl = encodeURIComponent(url),
      title = $this.attr('data-title'),
      tsina = $this.attr('data-tsina'),
      description = $this.attr('description');
  var html = [
  '<div class="hoverqrcode clearfix"></div>',
  '<a class="overlay" id="qrcode"></a>',
  '<a href="https://www.facebook.com/sharer.php?u=' + encodedUrl + '" class="article-share-facebook" target="_blank" title="Facebook"></a>',
  '<a href="https://twitter.com/intent/tweet?url=' + encodedUrl + '" class="article-share-twitter" target="_blank" title="Twitter"></a>',
  '<a href="#qrcode" class="article-share-qrcode" title="微信"></a>',
  '<a href="http://widget.renren.com/dialog/share?resourceUrl=' + encodedUrl + '&srcUrl=' + encodedUrl + '&title=' + title +'" class="article-share-renren" target="_blank" title="人人"></a>',
  '<a href="http://service.weibo.com/share/share.php?title='+title+'&url='+encodedUrl +'&ralateUid='+ tsina +'&searchPic=true&style=number' +'" class="article-share-weibo" target="_blank" title="微博"></a>',
  '<span title="Share to"></span>'
  ].join('');
  $this.append(html);

  $('.hoverqrcode').hide();

  var myWidth = 0;
  function updatehoverqrcode(){
    if( typeof( window.innerWidth ) == 'number' ) {
      myWidth = window.innerWidth;
    } else if( document.documentElement && document.documentElement.clientWidth) {
      myWidth = document.documentElement.clientWidth;
    };
    var qrsize = myWidth > 1024 ? 200:100;
    var options = {render: 'image', size: qrsize, fill: '#2ca6cb', text: url, radius: 0.5, quiet: 1};
    var p = $('.article-share-qrcode').position();
    $('.hoverqrcode').empty().css('width', qrsize).css('height', qrsize)
                          .css('left', p.left-qrsize/2+20).css('top', p.top-qrsize-10)
                          .qrcode(options);
  };
  $(window).resize(function(){
    $('.hoverqrcode').hide();
  });
  $('.article-share-qrcode').click(function(){
    updatehoverqrcode();
    $('.hoverqrcode').toggle();
  });
  $('.article-share-qrcode').hover(function(){}, function(){
      $('.hoverqrcode').hide();
  });
});   
</script>









<link rel="stylesheet" href="/typhoon/fancybox/jquery.fancybox.css" media="screen" type="text/css">
<script src="/typhoon/fancybox/jquery.fancybox.pack.js"></script>
<script type="text/javascript">
$(document).ready(function(){ 
  $('.article-content').each(function(i){
    $(this).find('img').each(function(){
      if ($(this).parent().hasClass('fancybox')) return;
      var alt = this.alt;
      if (alt) $(this).after('<span class="caption">' + alt + '</span>');
      $(this).wrap('<a href="' + this.src + '" title="' + alt + '" class="fancybox"></a>');
    });
    $(this).find('.fancybox').each(function(){
      $(this).attr('rel', 'article' + i);
    });
  });
  if($.fancybox){
    $('.fancybox').fancybox();
  }
}); 
</script>



<!-- Analytics Begin -->



<script>
var _hmt = _hmt || [];
(function() {
  var hm = document.createElement("script");
  hm.src = "//hm.baidu.com/hm.js?e6d1f421bbc9962127a50488f9ed37d1";
  var s = document.getElementsByTagName("script")[0]; 
  s.parentNode.insertBefore(hm, s);
})();
</script>



<!-- Analytics End -->

<!-- Totop Begin -->

	<div id="totop">
	<a title="Back to Top"><img src="/typhoon/img/scrollup.png"/></a>
	</div>
	<script src="/typhoon/js/totop.js"></script>

<!-- Totop End -->

<!-- MathJax Begin -->
<!-- mathjax config similar to math.stackexchange -->


<!-- MathJax End -->

<!-- Tiny_search Begin -->

<!-- Tiny_search End -->

  </body>
</html>
